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Кто я? 


Меня зовут Валерий Горбачев 
Работаю программистом в ОеЦуегу СшБ 
Использую: РНР, ]5, 50 


немного организатор (митапы по РНР в Краснодаре) 
немного спикер (локальные митапы) 

участник команды разработки УП3 Тгате\м/огК 
основатель чата @рНркга 


План доклада 


Исторический экскурс 
Грабли 
Костыли 


Вывод 


@ОАВКОЕР_РВ 


Ос айтег: 


Все ошибки случайны и связаны с плохой памятью докладчика, а не совершены 
по злому умыслу. 


Как я изучал БД 


Классическая кривая обучения: 
Отрицание 

Гнев 

Торг 

Депрессия 

Принятие 


о 
# .? 
® «? 


Как я изучал БД 


Класеичеекая-кривая-ебучения: 


Как было на самом деле: 


° Данные в файлах 
» Файловые БД (а, аВа$е №) 
° Изучение $ОЕ (книги, документация) 


# 


@ 


.# «? 
® 
.# 


Запросы к БД - это просто 


Я уже умею писать 5О!-запросы: 


$$91 = "ЗЕЁЪЕСТ * ЕВОМ ргодисЕ$ МНЕКЕ 19 = {$14}" 


и даже иногда забочусь о безопасности 


$$91 = 'ЗЕЁЪЕСТ * ЕВОМ ргодисф$ МНЕКЕ 19 = ' . @1пуа1($19); 


Не «само» развитие 


Когда мне приходилось сохранять строковые данные, я делал вот так: 
$591 = "ЗЕЁТЕСТ * ЕВОМ ргодисЕ$ МНЕВЕ паме = '{$пате}'"; 


однажды меня взломали, и я узнал про $ОЁ-уязвимости 


Не «само» развитие 


Когда мне приходилось сохранять строковые данные, я делал вот так: 


$$91 = "ЗЕЪЕСТ * РЕКОМ ргодисе$ МНЕКВЕ пате = '{$пате}'"; 


однажды меня взломали, и я узнал про $ОЁ-уязвимости 


Я начал делать вот так и делал так не один год: 


Фпаме = муза1_геа|1_е$саре_$Ег1лпд($памте); 
$$91 = "ЗЕЁЕСТ * ЕКОМ ргодисЕ$ МНЕКЕ пате = '". $пате. "'"; 


Потом узнал, что Му5ОМ лучше (когда ту$а[_” задепрекейтили) 


$5ЕтЕ = $ту$а11->ргераге ( "ЗЕТЕСТ * ЕВОМ ргодисЕ$ МНЕКЕ памте = ?"); 
$ъЕтЕ ->61п4_рагат( "$", $паме); 


Про дергесаеей туза! _ 


е РНР5.О — появился ехИтузаи (2005) 


Про дергесаееа туза. 


е РНР5.О — появился ехИтузаи (2005) 
е РНР 5.1 — появление модуля РОО_МуФОЕ (2005) 


Про дергесаееай туза! _ 


е РНР5.О — появился ехИтузаи (2005) 
. РНР 5.1 — появление модуля РОО_Му$Ои (2005) 
е РНР 5.5 — появление тузапа (2009) 


Про дергесаееа туза. 


РНР 5.0 — появился ехИтузаи (2005) 

РНР 5.1 — появление модуля РОО_Му5ОЕ (2005) 

РНР 5.3 — появление тузала (2009) 

РНР 5.* — переход с Цбтуза(сИепе (С++, Огаце) на тузала (2013) 


Про дергесаееа туза. 


РНР 5.0 — появился ехИтузаи (2005) 

РНР 5.1 — появление модуля РОО_Му5ОЕ (2005) 

РНР 5.3 — появление тузала (2009) 

РНР 5.* — переход с Цбтуза(сИепе (С++, Огаце) на тузала (2013) 

РНР 5.5 — модуль туза[_^ помечен устаревшим и удалён в версии 7 (2015) 


... В ЭТО время я задумался о переходе на ту5ОНИРОО (для старого кода) 


Рекомендовано для новых проектов 

АР! поддерживает асинхронные, неблокирующие запросы ту5апа 
Постоянные (регу1${еп) соединения 

АР! поддерживает подготовленные запросы на стороне сервера 
АР! поддерживает подготовленные запросы на стороне клиента 
Поддерживает всю функциональность Му5О( 5.1+ 


АР! поддерживает множественные запросы 


Му5ом 
Да 

Да 

Да 

Да 

Нет 

Да 

Да 


РБО_Му$Ое 
Да 

Нет 

Да 

Да 

Да 
Большинство 


Большинство 


Используя РОО, можно легко сменить используемую СУБД © нея 


к сожалению, я это запомнил... 


И на этом можно было бы остановиться... 


На новых проектах я использовал ОВМ из состава фреймворков 


ФизегОцегу = (пем Очегу()) ->зе1есЕ("'149')->Ргом( 'изег'); 


И на этом можно было бы остановиться... 


На новых проектах я использовал ОВМ из состава фреймворков 


ФизегОцегу = (пем Очегу()) ->зе1есЕ('149')->Ргом( 'изег'); 
На старых проектах я перешёл на Му5ОШ /РБО 
Му5ОП -— ведь это было максимально быстро и просто, но в биндинге можно 


легко запутаться. 


$$ ЕтЕ->61пЧ_рагам( "$$$$11$1$$1$5", $фпаме, $Е1 Те, ...); 


ОКМ - лекарство от всех проблем 


Основные преимущества ОКМ 


Код может быть абстрактным 

Нет нужды писать «сырые» запросы 
Порог вхождения гораздо ниже 

Код не слишком привязан к движку БД 


Е оачепт, Бостте, Суе, Ргоре|, Ке4ВеапРНР, УПОВ ит.д. 


Недостатки ОВМ 


Без обобщений — дальше я буду говорить, опираясь на опыт, полученный от 
использования ОКМ от Уп Егатемогк 1/2 


Ключевые недостатки 


Нет той же гибкости, как при написании «сырых» запросов 
Сложно писать сложные запросы 

Работает быстро, но немного медленнее нативных библиотек 
Иногда (едасу по принципу «так сложилось» 
Му5О-Яг${-подход 


Хождение по граблям как стиль жизни 


давайте погрустим вместе — мне мешают использовать ОКМ всю жизнь :( 


СИррег или приключения в сетях 1РХ/ЗРХ 

(++ ВиЦаег и Рагадох в мире блокировок 

Кто одержит победу: ту5а[ у$ туза 

Т-5О(. Бизнес-логика в хранимых процедурах 

Огацще + ОС. Взаимодействие через хранимые процедуры 
Инкапсуляция запросов в АР! с использованием М 5ОЁ 


Гарри Поттер и Орден Феникса 


Перемотка 


Пропустим затянутый очерк о моей жизни и перейдём к сути: 


Познакомился с Александром Макаровым 

Поковырялся в УП2 по части МУ 50 

Догадался, где пригодится поверхностный опыт использования разных БД 
Познакомился с УЛ (тег Агатб ща, и мы вместе рефакторим уп5оН/аь 

В основе пакетов у!50Н/ЧЬ лежит РОС... 


Скучный пересказ 
документации 


в которой можно найти ответы на большинство вопросов 


РОО как глоток свежего воздуха 
Подстановка данных может быть удобной... 
Здесь могла бы быть картинка 
с девушкой ведущей вас за руку 


Но презентация про РОО, 


поэтому без весёлых картинок 


Использование РОО 


В использовании РОО нет магии, и выглядит всё очень просто 


/* Выполнение запроса с передачей ему массива параметров */ 


$391 = 'ЗЕЁЕСТ паме, со{\оиг, са\1ог1ез$ 
ЕВОМ гит 
МНЕВЕ са|1огте$ < :са\1оглез АМО со{оцг = :со|о0иг'; 


$$ЕИ = $фабй->ргераге($$491, ['са\ог1ез' => 150, "'со1оиг' => 'геа']); 
$51 ->ехесиее (); 
$геЧ = $5Е1->РеЕсНнАТТ(); 


Пример взят из официальной документации 


Огромные возможности РОО 


РОРО_СУВЕ!Ю 
РОО ОВИВ 
РРО_ЕВЕВИВО 
РОО _!ВМ 
РРО_ИМЕОВМИХ 
РРО_МУ$ОЕ 
РРО_ОС! 

РОО _ООВС 
РОО _Рб$ОЕ 
РОО $ОНТЕ 
РОО _$О1$ВУ 


Сибма 

ЕгееТО5 / Мгсго5о В ОЕ $егуег / ЗуБазе 
Нгеб!га 

|ВМ 082 

ВМ пГогтих Эупаптис $егуег 

Му5ЗОЕ 3.х/4.х/5.х 

Огас‹е СаЦ {еГасе 

ООВС 3 (1ВМ ОВ2, итхОБВС и мт352 ОБВС) 
Розтдгез ОЕ 

5Ое 3 и 5ОЦЁе 2 

М!сго5ой 5О( $егуег / 501 Атиге 


Другой «движок» базы данных 


МубОЕ Роздге О 1 — отраслевой стандарт, который должен знать каждый 
уважающий себя у/еб-разработчик. 


Но это ведь не проблема, когда утебя РОО? 


Другой «движок» базы данных 


МуЗОЕ Розтдге О | — отраслевой стандарт, который должен знать каждый 
уважающий себя у/еб-разработчик. 


Но это ведь не проблема, когда утебя РОО? 


Но: 
1. Запрос запросу рознь (ЫМП 10 ОЕРЪФЕТ 10 у5 ЫМП 10, 10) 
2. Универсального кода не бывает 


5. Вообще — бывает, но работает неожиданно... 


обро пожаловать в жестокий мир РОО 


ТОП-5 


неиспользуемых режимов РОО: {еп 


Из документации... 


<?рйр 
$5ЕИ = ФаБп->ргераге ( "ЗЕТЕСТ пате, со{оиуг ЕКОМ Тги1{"); 
$51 ->ехесиее(); 


/* Извлечение всех оставшихся строк результирующего набора */ 

рглпЕ ("Извлечение всех оставшихся строк результирующего набора: \п"); 
$гези1Е = $$Е1->РеесвдТТ(); 

ргзпЕ_г($гези1 Е); 

э> 


Самые используемые: РЕТСН_А$$ОС или РЕТСН_ОВ] 


НЕТСН КЕУ РАВ 


ЗЕГЕСТ изегпате, етал1 ЕКОМ изег$ 


'1091п1' => 'ета111@та11.ги', 
'1091п2' => 'ета112@та11.ги', 


Просто Кеу-уаШше. Следит, чтобы в выборке было только два поля. Перезаписывает последним ключом. 


НЕТСН ИМЮЦЕ 


Сочетание: РВО::РЕТСН_УМЮЧЕ | РОО::;РЕТСН_А$$О0С 


ЗЕГЕСТ `14`, `ичзегпате`, `еталз1` ЕКОМ `изег$` 


1 => ['чзегпате' => '1091п1101', 'ета11' => 'ета111@та1лТ.ги', |], 
2 => ['изегпате' => '1091п1102', 'ета11' => 'ета112@та1Т.ги', |], 


Всё как и в предыдущем случае, только значения в виде массива. Следим за уникальностью сами. 


НЕТСН СВОИР 


Сочетание: РВО::РЕТСН_СКОУР | РВО::РЕТСН_А$$ОС 


ЗЕЕЕСТ `дгоир`, `14`, `изегпаме` ЕКОМ `изег$` 


[ 
'геадег' => [ 
['19' => 1, 'чзегпате' => '1091п101'], 
['19' => 3, 'изегпамте' => '1091п103'], 
], 
‘мгтеег' => [ 
['19' => 2, 'изегпате' => '1091п1102'], 
['19' => 4, 'изегпате' => '1091п1104'], 
], 
] 


А воттак — группировка по первому значению в выборке. 


ГЕТСН_МАМЕС вместо РЕТСН_А$5ОС 


ЗЕЕЕСТ и. `паме`, и. `етал11`, 9. `пате` 
ЕВОМ `изег$` и 


ТММЕВ ЭОТМ `дгоир$` д ОМ (9. `1а` = ци. `дгоир_1а` 


$$ те ->РеЕсй (РОО: : ЕЕТСН_МАМЕО) ; 
а 
'пате' => [ 
0 => '1091п101', 
1 => 'мгщег', 
], 
'ета11' => 'ета111@та1л1.ги', 
а 


А воттак мы не потеряем значение, если поле в выборке несколько раз. 


НЕТСН СЕА$$ТУРЕ 


ЭЕЕСТ 
СОМСАТ ( УСАЗЕ (ЕЕРТ(`дгоиур`, 1)), ЗУВЗТВТМС(`дгоир`, 2)), 
`изегпаме`, `етал1` 

ЕКОМ `изег$` 


$$ ЕщЕ ->Реесй (РОО: : РЕТСН_СЁА$З$ | РОО: : РЕТСН_СЕАЗЗТУРЕ); 
ит 
об]есЕ (Кеадег)#3 (1) { 
["чзегпате"] => $&г1п9(8) "10911101", 
["ета11"] => $&г1п9(14) "ета111@та11.ги", 
ро 


Пахнуло дьнмем-езаведа фабрикой? А будет ли это работать в РНР9? 
Интересно? Воттут детально: ПЕрз://рпраеи1оп$.пе/рао/ЛесП_тойе$ 


А есть ли еще «занимательные флаги» 


° РОО::АТТВ_ОКАСЬЕ_МУЕЕ$ — "" => МУ, МОШЕ => "" (не только Ога) 
» РОО::АТТВ_САЗЕ — регистр имён столбцов (Му5ОЕ 8.0.21) 
° МУЗОЕ АГТК_1ОСАЕ 1МЕШЕ_ОВЕСТОКВУ (с РНР 8.1) 


ГОА ВАТА ГОСАЁЕ ТМЕТЬЕЕ 'раЕП/Ео/Р11е/Р11е. ЕхЕ' 


А есть ли еще «занимательные флаги» 


РОО::АТТК_ОКАСЬЕ МУ 5$ — "" => МУ, МИЕЕ => "" (не только Огасе) 
РОО::АТТВ_САФЕ — регистр имён столбцов (МубОи 8.0.21) 
МУ$ОЕ_АТТВ_1ОСАЕ 1МЕШЕ_ОВЕСТОКУ (с РНР 8.1) 


ГОА ВАТА ГОСАЁЕ ТМЕТЬЕЕ 'раЕП/Ео/Р11е/Р11е. ЕхЕ' 


Странные флаги: 
РОО::МУЗОЕ_АТТК_МИУЕТЕ$ТАТЕМЕМТЬ 
РОО::АТТК_РЕТСН_САТАЕОС МАМЕЗ 
РОО::АТТК_РЕТСН_ТАВЕЕ_МАМЕЗ 


Мне кажется, что РОО ждёт большое будущее... БгеаКтд сВапде в РНР 9 


Регистр и Му5ОЕ 8.0.21 


Что можно ожидать в результате выполнения запроса? 
ЗЕЕЕСТ `сопз&галп_паме` Тгот `1пРогма&1оп_зсПпета`. `Кеу_со1итп_изаде`; 
ух 
Му$ОоЕ < 8.0.21 Т1е1\4$ пате: 


аггау(1) { 

'сопзЕга1пЕ_пате' => "пате_оР_сопЕгалпЕ" 
} 
МузоЕ 8.0.21 Г1е1а$ паме: 
аггау(1) { 

'СОМ5ТВАТМТ_МАМЕ' => "пате_оР_сопЕгалпе" 
} 
а 


И воттут на сцену может выйти: РОО::АТТК_САЗЕ => РОО::САЗЕ 1О\М/ЕК 
ЮЕр5://аипиб.сот/уй$оН/уй2/155$4е$/18171 


Хранение )5О0М в ВЕОВ 


$$91 лпзегЕ = <<<$01 
1п5зегЕ 1пео `р4о_урез`(`61о6_со1`) уаШшез( : 61об_4ата); 
$0Е; 


$591_геаЧ = <<<$01 


зе\есЕ `610о6_со1` Тгот `рдо_фурез$`; 
01; 


“ сейчас будем делать примерно как в документации 


“и да, мы не используем СЁОВ, ]}5ОМВ и т.п. спец.типы для хранения 


Му5О| 


Вставляем данные 


$рдо->ргераге ( $$91_1пзеге ) ->ехесиЕе( [':61об_Чафа' => $01обрака]); 


Читаем данные 
$зЕтЕ = $рдо->дчегу ( $$491_геаа); 
$гези1Е = $5ЕтЕ->Реесй (РБО: : РЕТСН_СОЕУММ); 
уаг_аитр ($гези1{); 


Радуемся 


Роздге ОЕ 


Вставляем данные 


$рдо->ргераге ( $$91_1пзеге ) ->ехесиЕе( [':61об_Чафа' => $01обрака]); 


Читаем данные 
$зЕтЕ = $рдо->дчегу ( $$491_геаа); 
$гези1Е = $5ЕтЕ->Реесй (РБО: : РЕТСН_СОЕУММ); 
уаг_аитр ($гези1{); 


Удивляемся 
гезоигсе(9) офт туре (5зЕгеамт) 


Роздге ОЕ 


Вставляем данные 


$рдо->ргераге ( $$91_1пзеге ) ->ехесиЕе( [':61об_Чафа' => $01обрака]); 


Читаем данные 
$зЕтЕ = $рдо->дчегу ( $$491_геаа); 
$гези1Е = $5ЕтЕ->Реесй (РБО: : РЕТСН_СОЕУММ); 
уаг_аитр ($гези1{); 


Удивляемся 
гезоигсе (9) оф туре (5зЕгеамт) 


Читаем документацию и радуемся 
уаг_Читр ($ геам_деЕ_сопЕепЕ$ ($гези1{)); 


Немного экзотики — Ога е 


Вставляем 


$рдо->ргераге ( $$91_1пзеге ) ->ехесиЕе( [':61об_Чафа' => $01обрака]); 


Удивляемся 
РНР Рафа1 еггог: ЦпсачдпЕ РООЕхсерЕ1оп: ЗОЕЗТАТЕ[НУ000]: бепегат 
еггог: 1465 ОСТЗЕМЕЕхеси*е: ОКА-01465: 1пуа114 Пех питбег 


Читаем документацию 


В случае с базами Ога{е требуется несколько иной синтаксис для извлечения 
содержимого файла и помещения в базу. Также необходимо выполнять вставку в рамках 
транзакции, иначе вставленный [ОВ будет зафиксирован в базе с нулевой длиной, так 
как если не обозначить границы транзакции, изменения будут фиксироваться после 
каждого выполненного запроса. 


Работающий код для вставки 


$591 лизегЕ = <<<$01 

1п5зегЕ 1пео "р4о_урез" ("Б1об_со1") уаШшез$ (етрЕу_61о6()) гебигп1по 
"отОб_соЕ" ТАЕО ББ юЮб-даЕа 
$0Е; 


$Гр = Тореп( 'рИр://тетогу', ‘гмЬб’); 
Рмг16е($РГр, $01обрака); 
Рзеек($Ер, 09); 


$$ЕтЕ = $рдо->ргераге ( $$91_1пзегЕ); 
$рао->6е91пТгапзасЕ1оп(); 
$$ ЕтЕ ->61п\а1ие(' :61о6_Чафа', $Рр, РОО: : РАВАМ_ЕОВ); 
$$ ЕтЕ->ехесиее(); 
$рао->сотм1 (); 


Вемипа? 


$5ЕтЕ = $рдо->дчегу ( $$491_геаа); 
$ЛапЧ1е = $$Ете->РеЕсИ (РОО: : РЕТСН_СОШОММ); 


ЗСОПЕеЙЕа: =", 
иуй11е (!Реот ($Папа1е)) { 
$сопфепе$ .= РТгеад ($ПапаТе, 2); 
} 
гемлпа ($ПапаТе); 
ий11е (!Реот ($Папа1е)) { 
$сопфепе$ .= РТгеад ($ПапаТе, 2); 


Работает и геумпа и {сеек 


Чиним ВЕОВ поля и Ога е 


Если данные у вас не совсем бинарные, к примеру /5ОМ/Ба5е64, то можно без 
транзакций и файлов. 


$5ЕтЕ = $аб->ргераге ("ТМЗЕВТ ТМТО {1(610о6_со1) 
МАГИОЕ$ (ТО_ВЕОВ(УТЕ_КАМ. САЗТ_ТО_КАМ( : 6 106_со1)))"); 
$ ЕтЕ ->61паРагам( ' :6106_со1', $61оббака); 


* лучше, конечно, хранить Базеб4 в С1ОВ, а |50п в ]50М 


Чиним ВЕОВ поля и Ога е 


Если данные у вас не совсем бинарные, к примеру /5ОМ/Ба5е64, то можно без 
транзакций и файлов. 


$5ЕтЕ = $аБ->ргераге ("ТМЗЕВТ ТМТО {1(6106_со1) 
МАГИЕ$ (ТО_ВЕОВ(УТЕ_КАМ. САЗТ_ТО_КАМ( : 6 106_со1)))"); 
$ ЕтЕ ->61паРагам( ' :6106_со1', $61оббака); 


* лучше, конечно, хранить Базеб4 в С1ОВ, а |50п в ]50М 


Изучаем особенности РОО дальше? 


Получение последнего вставленного |0 


Что может быть проще? 


Пример взят из официальной документации 


рчб11с РОО: : Таз ТизегЕ ТО (?$Ег1пд $пате = пи11Т): $зЕг1пд | Га1$е 


Также как и это замечание 


Замечание: 

В зависимости от драйвера РБОО этот метод может вообще не выдать 
осмысленного результата, так как база данных может не поддерживать 
автоматического инкремента полей или последовательностей. 


М$ 501 $егуег 


Гат гипттд ап тей диегу изутд РОО апа Пеп дейтд {пе пеуму сгеаеа Га ий 
[45 п5ег!а(). ГРи$ 15 а ииогкта оп ту [оса[По5$Е епигоптепеЕ. 

И/еп ! тоуе {Те ехасЕ 5ате соае опто а 5егуег, те [а5Ипзег!а( 15 амауз геигптд 
Ыапк, еуеп Поидй {те тзет уиетепЕ и/огк$ апа тзеп5 {Пе пеми гоми то те 


аавара$е. 
ПИр5://таскоуеПо\м/.сот/диеНоп5/15747481/рао-|[а5Итзега-по{-м/огКтод-Гог-п5-$4( 


М$ 501 $егуег 


Гат гипттд ап тей диегу изутд РОО апа Пеп дейтд {пе пеуму сгеаеа Га ий 
[45 п5ег!а(). ГРи$ 15 а ииогкта оп ту [оса[По5$Е епигоптепеЕ. 

И/еп ! тоуе {Те ехасЕ 5ате соае опто а 5егуег, те [а5Ипзег!а( 15 амауз геигптд 
Ыапк, еуеп Поидй те тзет уиетепЕ и/огК$ апа тзеп5 {Пе пеми гоми то те 


ааеара$е. 
ПИр5://таскоуео\м/.сот/дие$Ноп5/15747481/рао-[а5Итзега-по{-м/огКтод-РГог-п5-$4( 


ВКетагК$ 


Поддержка РОО была добавлена в версии 2.0 Драйверы МгсгозойЙ $ОЕ $егуег для РНР. 

Между версиями 2.0 и 4.3 необязательным параметром является имя таблицы, а возвращаемым значением — 
идентификатор последней добавленной в указанную таблицу записи. Начиная с 5.0, как необязательный параметр 
рассматривается имя последовательности, а как возвращаемое значение — последовательность, которую 
добавили для указанного имени последовательности последней. Если имя таблицы указано для версий после 4.3, 
1аз{ТпзегЕта возвращает пустую строку. Последовательности поддерживаются только в $С( $егуег 2012 и более 


поздних версиях. 


Смириться или приложить подорожник? 


Починим (аз тега для М!сго5ой ОГ Фегуег (старых версий) 


сТаз$ м$5а1РОО ехЕеп4$ \РОО 
| 
у** 
* КеЕигпз$ уа\ие ог {Пе ТазЕ 1пзегееа т). 
* @рагат $Еглпд |пи1Т $з$едчептпсе {Пе зедиепсе паме. ВеГаи1{$ Фо пи\1. 
* @гесигп 1пЕ ТазЕ 1пзегЕеЧ ТО уаше. 
т 
руб 11с Гипсё1оп ТазЕТпзегЕта ($зедицепсе = пи\1.) 
{ 
$$а1 = 'ЗЕЁЕСТ САЗТ ( СОАЕЕЗСЕ (ЗСОРЕ_ТОЕМТТТУ(), @@ТОЕМТТТУ) А$ 
О9тИЕ)“; 
геЕигп $Е01$->дчегу ( $$а1) ->РеЕсйСоТитп (); 


О15саттег 


Решение по излечению [а5Ипзег а может работать неожиданно... 


Гог ехатри(е, {Пеге аге {мо а ($, Т1 апа Т2, апа ап 1М5ЕКТ Етддег 15 аерпеа оп Т1. 
И/Реп а гоми 15 тзепеа то Т1, те таддег ргез апа тзег5 а гоми т Т2. ТР/5 $сепато 
ШизгаЕе$ мо 5сорез: Те т5еп оп Т1, апа {те тей оп Т2 Бу {пе {пддег 


ВЕръ://40с$.пп!сго$ой.сот/ги-КУ/5а/-$а9ИТипсНоп$/соре-1шепу-гапзас!-59(?\ме\м/=54|-егуег-2016 


|ОЕМТ_СУВКЕМТ — геёигп$ {Пе [а${ ету уаше депега{е4 ог а рес! Яс {аЩе т апу $е5$1оп ап@ апу 5соре. 
@(@РЕМТИУ -— гефигп$ {Пе [а5{ 14еп{Ку уаШе депегакеа Гог апу таЩе т {Пе сиггепт 5е$51оп, асго$$ аЦ 5сорез. 
$СОРЕ 1РЕМТИТУ -— гефигп$ {Пе [а5{ 14еп{Ку уаШе депегакеа Гог апу таЩе т {Пе сиггеп{ 5е$$1оп апа {Пе сиггеп{ 


5соре. 


Не используйте [а5Нпзег а? 


$р901 - >ргераге ( $$а1_1пзегЕ) ->ехеси*е([':п' => 101]); 
$р90о2->ргераге ($$а1_1пзегЕ) ->ехесие([':п' => 102]); 


еспо 'ТазЕТпзегЕТа1 = '.уаг_ехрогЕ ($р401 ->ТазЕТпзегЕта(), Егие).РНР_ЕОЁ; 
еспо 'ТазЕТпзегЕТа2 = '.уаг_ехрогЕ ($р4до2 ->1азЕТпзегЕтТа(), Егие).РНР_ЕОЁ; 
// ТазеТпзегЕ Та = '1' 
// ТазЕТпзегЕТтТа2 = '2' 


Му5ОЕ, РозгадеЪО(, М5 501 — РАЗЗЕБ 
Огацще — ЕАШ 


РНР Рафа1 еггог: ЦпсачдпЕ РООЕхсерЕ1оп: ЗОЕЗТАТЕ[ТМ001]: Оглуег 4оез пое 
зиррогЕ ЕП1$ РГипсЕЁ1оп: Чглуег 9ое$ поЕЁ зиррогЕ ТазЕТпзегЕТтТа() 


Но если очень хочется... 


Проверено с рПр8.0 + Огаце ХЕ 119 г2 + т5агсЦептЕ21_5 


$$(тЕ1 = $рд01 ->дчегу ( 'ЗЕЪЕСТ "р4о_урез_$ЕО".СУВКУАЕ ЕКОМ БУАЁ'); 
$гези1{1 = $56тЕ1 ->РеесИСоТитп (); 


$$ЕтЕ2 = $рдо2->дчегу ( 'ЗЕЪЕСТ "р4о_урез_$ЕО".СУВКУАЕ ЕКОМ БУАЁ'); 
$гези1{2 = $5(тЕ2->РеЕсИСоТитп (); 


Ялта = 1 
у/ тез = *2` 


Такой же костыль для М$ 501 


Проверено с рПр8.0 + 5ОЕ Фегуег 2017 (таблица без триггеров) 


$$ЕтЕ1 = $ра01 ->дчегу ( 
'ЗЕЁЕСТ САЗТ (СОАТЕЗСЕ ( ЗСОРЕ_ТОЕМТТТУ(), @@ТОЕМТТТУ) АЗ Бои е) ); 
$гези1{1 = $56тЕ1 ->РеЕсИСоТитп (); 


$$ЕтЕ2 = $рао2 ->ачегу ( 
'ЗЕТЕСТ САЗТ ( СОАЕЕЗСЕ ( ЗСОРЕ_ТОЕМТТТУ (), @@ТрЕМТТТУ) АЗ БТИ Е)°.); 
$гези1{2 = $56тЕ2->РеЕсИСоТитп (); 


я гаи = "1 
уд ГВЕН = "2 


Но будет работать и со старыми версиями. 


Непрошенный совет 


И всё же не используйте [а тзег а, есть способы лучше: 


Роге о — КЕТОКМИМС "1а" 

М$ 50 — ОЧТРИТ 1МФЕКТЕО.!О * 

Огац{е — ВКЕТУКМИМ С га МТО па 

МапарВ (с версии 10.5.0) — КЕТОКМИМС 11а` 
Му5ОЕ старых версий — никак 


Только используя РОО с Огаце, я понял, зачем нужно БпаРагат 


Еще немного непрошенных советов 


» РОО::МУ$ЗОЕ_ АТТВ 1МПТ_СОММАМО можно использовать для ЕТ МАМЕФЗ в 
сочетании с спаг5Е=и{8тЬ4 в ОЪМ строке 


Еще немного непрошенных советов 


» РОО::МУ$ОЕ_ АТТВ 1МПТ_СОММАМО можно использовать для ЕТ МАМЕФЗ в 
сочетании с спаг5®(=и{8тЬ4 в ОЪМ строке 
» Принудительно выключайте РВО::АТТВ_ЕМУЁЕАТЕ_РКЕРАКЕ$ 


Еще немного непрошенных советов 


» РОО::МУ$ОЕ_ АТТВ 1МТ_СОММАМО можно использовать для ЕТ МАМЕЪЗ в 
сочетании с спаг5®(=и{8тЬ4 в ОЪМ строке 

» Принудительно выключайте РВО::АТТВ_ЕМУЁЕАТЕ_РКЕРАКЕ$ 

° Именованые подстановки подставляются один раз (Му5ОЕ 1) 


Еще немного непрошенных советов 


РОО::МУЗОЕ_АТТК_1МТ_СОММАМОЬ можно использовать для 5ЕТ МАМЕФ в 
сочетании с спаг5Е=и{8тЬ4 в ОЪМ строке 

Принудительно выключайте РВО::АТТВ_ЕМУЕАТЕ_РВЕРАКЕ$ 

Именованые подстановки подставляются один раз (Му5ОЕ 1) 

Не ленитесь: ота\аше(‘акг’, $уаше, РОО::РАВАМ ПМТ) лучше ехесще(['‘аиг' => $уаше]) 


Еще немного непрошенных советов 


РОО::МУЗОЕ_АТТК_1МТ_СОММАМОЬ можно использовать для 5ЕТ МАМЕФ в 
сочетании с спаг5Е=и{8тЬ4 в ОЪМ строке 

Принудительно выключайте РВО::АТТВ_ЕМУЕАТЕ_РВЕРАКЕ$ 

‚ Именованые подстановки подставляются один раз (Му5ОЕ 1) 

‚ Не ленитесь: ыта\Уаше( аи, $уаше, РБО::РАВАМ ИМТ) лучше ехесще(['айг' => Фуаше]) 
МУ$ОЕ_АТТВ_У$Е_ВУЕРЕКЕО_ОЧЕВУ=Ра[$е для больших запросов (туза!па) 


Еще немного непрошенных советов 


РОО::МУЗОЕ_АТТК_1МТ_СОММАМОЬ можно использовать для 5ЕТ МАМЕФ в 
сочетании с спаг5®(=и{8тЬ4 в ОЪМ строке 

Принудительно выключайте РВО::АТТВ_ЕМУЕАТЕ_РВЕРАКЕ$ 

Именованые подстановки подставляются один раз (Му5ОЕ 1) 

Не ленитесь: ота\Уае(‘ак’, $уаше, РОО::РАВАМ ПМТ) лучше ехесще(['‘аиг' => $уаше]) 
МУ$ОЕ_АТТК_У$Е_ВУЕРЕКЕО_ОЧЕВУ=Ра[$е для больших запросов (туза!па) 
Избегайте многострочных 5О|-запросов в одном вызове. Но если уж так 


получилось: 
° ехесще() сообщит об ошибке только в первом запросе 
» Используйте пех{Вом/5е{() для выборки следующего набора 
» пех{Ко\м/5еТ() — может вернуть пустой результат, который нельзя выбрать (со\итпСоицп\()) 


Про пех{Вом5е(() 


Иногда не получается избежать многострочных запросов 

$591 лпзегЕ = <<<$01 

ЗЕТ М№ОСОиМТ ОМ; 

РЕСГАКЕ @+тр ТАВЕЕ ([19] 1пе); 

ТМЗЕКТ ТМТО [+аб1е]([\а1]) ОУТРУТ ТМ№$ЗЕВТЕО.19 ТМТО @Етр МАШОЕ$('12'); 


ЗЕЕЕСТ * РЕКОМ @етрогагу_1пзегЕед; 
ОЕ 


$5ЕтЕ = $рдо->ачегу ($$491_1пзеге); 


Чо { 
$гези1Е = $5ЕтЕ ->РеЕси (РОО: : РЕТСН_А$$0С); 
еспо 'со\итпСоипЕ = '. уаг_ехроге ( $$ ЕмЕ ->соТитпбСоипЕ (), Егие) . РНР_ЕОЦЫ; 
еспо 'гези1{ = ' . уаг_ехроге ($гези1{, (гие) . РНР_ЕОЕ; 


} мп11е ($5$ЕтЕ->пехЕКомзее ()); 


Временная таблица — особенность работы ОЧТРОТ 1М$ЗЕКВТЕР для таблиц с триггерами. 


Результаты гом/5е ов 
ОЕ$ВУ 


со1итпСоипЕ = 1 
гези1{ = аггау ( 
71а" => а 


155 ие, связанные с поведением: 


ОВЫВ 


со|1итпСоипЕ = 09 
гези[{Е = ТГа1зе 


со|1итпСоипЕ = 09 
гези1Е = Га1!зе 


со|1итпСоипЕ = 1 
гези1{ = аггау ( 
== 5, 


нет гом/5е{ только от ОЕСЕАКВЕ 


Соитп гом/5е{$ {0 Бе $КрреЯ ащотайсаЦу | Етр\у гези{ 5е{ героцеа Тог ТО 1МЕО те55аде$ 


А вы знали, что... 


Используя РОО + официальный драйвер от М!сго5ой, вы не сможете вставить в 
запрос больше 2100 значений с помощью ЫШпа\а(ше. 


РООЕхсер*1оп: $О1$ЗТАТЕ[ТМ$$Р]: Тглеа фо 61п рагамтефег питбег 2101. 
50Е 5егуег зиррогЁ$ а тахлтит оГР 2100 рагамееег$. 


зоигсе/5пагей/соге $а[гу.П 


172 соп$ф 1пЕ $50 $ЕКУЕК_МАХ_РАКАМ$ = 2100; 

173 соп$ЕЁ 11 $50Е_5ЕК\УЕК_МАХ_МОМЕУ_ $САЕЕ = 4; 

174 

175 1псгеазе {пе тах1тит теззаде 1епдЕй То ассоттода{е Гог {пе 1о0пд еггог геГигпед Тог 
орегапа Туре с1а$в 


ИЕр5://9Пиб.сот/пписгозоН/тзрИрз41$$ие$/410 


Именно это часто толкает к использованию ОВИВ 


— Не ходи туда, там тебя ждут неприятности. 


— Ну как же туда не ходить? Они же ждут! 


© Котёнок по имени Гав 


Еще мелочей? 
Что может пойти не так при обычном ЗЕСЕСТ? 
РНР 7.4/8.0 + РБО (Му$ОЕ) 


// ЗЕЁЕСТ * ЕКОМ $14тр1е_фаб1е МНЕВЕ 19=1 
уаг_аитр ($гези1{); 


р 

аггау(4) { 
ео => $6г1п9(4) "-123" 
‘ото пЕ со [” == 5219 (10) 9817906572” 
ГТОЕ СОЕ" => $6г1п9(11) "-12345.6789" 


'путег1с_со1' => $&г1п9(6) "-33.22" 


} 
г. 


Обновляем версию РНР 


РНР 8.1 + РОО (Му5ОЕ) 


// ЗЕЁЕСТ * ЕКОМ $14тр1е_фаб1е МНЕВЕ 19=1 


уаг_Читр ($гези1{); 


и” 


аггау(4) { 


} 
г. 


[ тПЕ 60" ] 
[Вот Е Со" 
Е о: Я 
["пимег1с_со1"] 


=> 


ИЕ (123) 

112 (8817806877) 
ТТоаЕ (-12345.6789) 
$Ег1пд9(6) "-33.22" 


Ая хочу как раньше 
РНР 8.1 + РОО (Му$01) 
$рао = пем РОО('ту$а1:...... ‚ спагзее=иЕР8', 'изегпате', "'раз$мога'); 


// для версии 8.1 

$рао ->зеЕАЕЕглрбиЕе (РОО: : АТТК_ЗТВТМ6ТЕУ_РЕТСНЕ$З, 1); 

// в драйвере до версии 8.0 флаг включен. С 8.1 флаг не влияет. 
$радо ->зе{АЕЕглБиЕе (РОО: : АТТК_ЕМУЕАТЕ_РКЕРАКЕ$, 1); 


и 

аггау(4) { 
ИЕ СО => $6г1п9(4) "-123" 
'61921пЕ_с0о1' => $6г14п9(10) "8817806877" 
`ГТОЕЕ СОЁ' => $6г1п9(11) "-12345.6789" 


`ПЫМегтТсС_ СО” =>. $27110 (6) "-33.22" 


| 
— 


Удовлетворим любопытство? 


Было 


#1Е ЭТУЕОЕ_7ЕМО_ГО№6==4 


1 ((164(2147483647) < (1п164_&) 1уат) || (164(-2147483648) > (1п164_®) 1\а1)) { 
ОВб_ТМЕ("$Ег1пда1Ру"); 


{тр_Т1еп = зрилпЕТ((сПпаг *)&&тр, "%" РКТ164, 1уаТ); 


} е!зе 
Н#епа1Р /* $ТИЕОЕ */ 
{ 


ГМАЁЕ_1ОМб(7\, (2епд_1опд) 1\а1т); /* (Пе сазЕ 15$ заГе, ме аге 1п {Пе гапде */ 


1 ((тр_Теп) { // ВОТ ЭТО МЕСТО 
МАЕ _ЗТВТМСЕ(72\у, тр, Етр_1еп); 


Удовлетворим любопытство? 


Стало 


#1Р ЭТИЕОЕ_ХЕМО_ГОМб==4 
1Р ((164(2147483647) < (1п164_&) 1уат) || (164(-2147483648) > (1п164_е) 1\а1)) { 
ОВб_ТМЕ("$Ег1пд1Ру"); 
ГМАЕ_$ЗТВ(2\, 2епа_164_о_$Ег(1\уа1)); 


} е!зе 
Н#епа1Р /* $УТИЕОЕ */ 
{ 


ГМАЕ_1ОМб(7\, (72епд_1опд) 1\ат); /* (Пе сазЕ 15$ заГе, ме аге 1п {Пе гапде */ 


} 


Удовлетворим любопытство? 


1тр_Теп = зрг1п+((спаг *)&{тр, "%" РВТ164, 1уа\); ТМАЕ_5ТВ(2\, гепд_164_10_$1г(1уа1)); 
} е\зе } е\зе 
#еп91+ /х* $Т2ЕОЕ */ #еп91+ /* $Т2ЕОЕ х, 
4 4 
ТМАЕ_ГОМб(2\, (2гепд_1опд) 1уа1); /х Те са$Е 15$ зате, и ТМАЕ_10№6(2\, (2еп9_Топд) 1уа1); /х тПе саз 
} } 
} } 
2+ (Етр_1еп) 4 (*гом)+= Буфе_соуп{; 
ТМА _5ТЕТМОЕ(2\у, тр, 1тр_Т1еп); О0В6_\УОТО_ВЕТУВМ; 


} 


(*гош)+= Буфе_соупт; 


СКВапдез опу 


Мег5юп ОаЁе АцКог Соттй Мез5аде 


65а5с18 13.04.2021, 16:43 АЗЗ ГипсНоп$ Го сопуегЕ 164/и64 Го $ пд 


Слабая динамическая типизация 


ес(аге ($1 Курез=1); 


с1аз$ ТурезСпеск { 
ри1уаЕе 1пЕ $п = 1; 


риб11с РГипсётоп сВеск(РОО $рдо) 
{ 
$зЕтЕ = $рао->ачегу ( 'зе1есЕ 1пЕ_со1Т Тгот рд9о_урез'); 
$$ ЕтЕ ->61паСо1итп(1, $Е01$->п, РОО: : РАВАМ_ТМТ); 
$Е11$ ->апумогк ($Е01$->п); // 1пЕ(1) 
$$ ЕтЕ ->РГеесй (РОО: : РЕТСН_ВОУМ№) ; 
уаг_Читр ( $Е11$->п); // $Ег1п9(3) "101" 
$Е11$ ->апумогк ($Е11$->п); 


// РНР Рафа1 еггог: Цпсаидйе ТуреЕггог: ТурезСПеск: :апумогк()... 


} 


рг1\уаёе ГипсЕ1оп апумогКкК(1пЕ $уа\ие) { /* апу асЕ1оп$ */ } 


} 
РОО::АТТК_ЕМОЕГАТЕ_РКЕРАВЕ$ = 1 


Транзакции в РОО 


» Следите за РВО::АТТВ_ЕВВМОВЕ (РОО::ЕКВМООЕ_$1ЕМТ, ес.) 


Транзакции в РОО 


е Следите за РВО::АТТВ_ЕВКМОВЕ (РОО::ЕВКМОБЕ $ ЕМТ, ес.) 
® ОО в транзакции поддерживают не все ОВМЪ — будет автокоммит (МуЗОЕ,, 
Огаце, М5 $01) 


Транзакции в РОО 


» Следите за РВО::АТТВ_ЕВВМОВЕ (РОО::ЕКВМООЕ_$1ЕМТ, ес.) 


® ОО в транзакции поддерживают не все ОВМЪ — будет автокоммит (МуЗОЕ,, 
Огаце, М5 501) 


е При старте РОО только проверит, что драйвер их поддерживает и нет активной 
транзакции 


Транзакции в РОО 


» Следите за РВО::АТТК_ЕКВМОВЕ (РОО::ЕККМООЕ $ ЕМТ, е{с.) 

® ОО в транзакции поддерживают не все ОВМЪ — будет автокоммит (МуЗОЕ,, 
Огаце, М5 501) 

е При старте РОО только проверит, что драйвер их поддерживает и нет активной 
транзакции 

° Вложенные транзакции не поддерживаются” 


“Свои методы для проверки активности транзакции *_ Пап е_т_{гапзасйоп имеют 
только Роге ОЕ и МуъОЕ (РНР 8.0), и они узнают о транзакции, даже если вы 
стартовали её через 501. 


Вложенные транзакции 


ТР (р9о_1$_1п_ЕгапзасЕ1оп(а6п)) { 
гепа_{Игом_ехсер1оп_ех(рИр_р49о_деЕ_ехсерЕтопт(), 0, 
"ТВеге 1$ а1геаду ап асЕ1уе фгапзасЕ1от"); 
ВЕТУВМ_ТНКОМ$ (); 


| 


зтаЕ1с 6001 р4о_1$_1п_ЕгапзасЕтоп (р4о_абп_Е *а6п) { 
1г (абп->теЕпо9$->1п_ЕгапзасЕтоп) { 
геригп абп->теепо4Ч$->1п_ЕгапзасЕтоп (а6п); 
} 


гесигп АБП->1п_&хп; // Боо1еап флаг 


3 


А вот метод из условия: аАБП->те{о@5->т {гапзасНоп 


зтаЕ1с Боо1 р4о_му$491_1п_Егапзасе1оп(рд9о_а6п_Е *а6п) 
С 
рао_ту$а91_Ч6_ПапЧ9Те *Н = (рдо_му$91_Ч6_Папа1е *)абп->9глуег_дака; 
РБО_ОВС_ЕМТЕК ( "р9о_ту$491_1п_ЕгапзасЕ1оп"); 
РБО_ОВС_РЕТУКВМ( (р4о_муза1_деЕ_зегуег_$фафи$(Н->$егуег) & ЗЕКУЕК_$ЗТАТУ$_ТМ_ТКАМЗ) != 0); 
} 


Транзакции и МУ5АМ 


$р901 - >Бед1пТгапзасЕ1оп(); // {гие 
$1пзегЕКези11 = $р9о1 ->ргераге ( $$91_1пзегЕ) ->ехесие([':п' => 111]); 
$р901->го11Васк(); // {гие 


$5Е = $рд01 ->ргераге ( 'ЗЕЁЪЕСТ 14, 1пЕ_со1 ЕВОМ туре МНЕВЕ 1п%_с01=:п'); 
$$Е->ехеси*е( ['п' => 111]); 
уаг_Читр ($$Е->Геесй (РОО: : Е+ЕТСН_А$$0С)); 


аггау(2) { 

'14' => $&г1п9(1) "3" 

‘ПЕ СО" => 56Га9(3) "1" 
} 


Документация обещает исключение в таких случаях, но мы его не получим, т.к. 
драйвер ничего не знает про Му15АМ-таблицу, ведь БД у нас ИММОВВ. 


Транзакции в М55 О 


С использованием ОВЫВ (ЕгеетО5 апуег Рог М!сгозой $О( егуег апа 5убазе да{аБазез) 


$р4о = пем РОО('9$п_$Ег1пд', 'изегпате', 'раззмога'); 
$рао->6е91пТгапзасЕ1оп(); 

$р9о->сотмт (); 

$р9о->го11Васк(); 


СрИр5-зуБа$е (о1а апуег Гог М$ 5О1 5егуег) 


$р4о = пем РОО('9$п_$Ег1пд', 'изегпате', 'раззмога'); 
$рдо->ехес ( 'ВЕСТМ ТКАМЗАСТТОМ'); 

$рдо->ехес ( 'СОММТТ ТВАМЗАСТТОМ'); 

$рдо->ехес ( 'КОГЕВАСК ТКАМЗАСТТОМ'); 


Лучше, конечно же, использовать РОО_$О15$КУ, но посмотрите, какие чудесные «костыли». 
Аеще так можно реализовать вложенные транзакции... 


История про РОО::диое и ;' 


[а{п1, №'5 {Пе $гт9д 2. Моте {Па т [а{т1 апа оаБК, 0х27 оп ($ ом/п 15 а Шег 


сПагасеег. 


ТНе рау!оаа ме'ге дотпд о изе ог {15 туесНоп °кагЕ$ УМЕН {Пе Буе зедиепсе пре |п ОБК, {ПаЁ$ ап пуаИ тийБуее спагасеег; п 
а 


ехИрао_тузаИтуза! Апуег.с 


1Г (изе_паЕ1опа\1_сПпагас®ег_зе{) { 
*дцотеЧ1еп = му$91_геа1_езсаре_$Ег1пд_адчо*е(Н->зегуег, 
ипацотедТет, '\''); 
(*ацофеа)[9] = 'м№'; 
(*ачцоЕеЧч) [1] = '\''; 


++* дциотед1епт; /* М ргеР1х */ 
} е\зе { 
*ЧцоЕеЧ1ептп = ту$41_геа1_езсаре_$г1пд_дичоее(Н->5егуег, 
ипацотедТет, '\''); 
(*ацофеа) [9] = '\''; 


(*диотечд) [++*диотеа\еп] 
(*диотечд) [++*диотеа\еп] 


А: 
'\0'; 


*дацофечд + 2, ипацотеа, 


*дцотеа + 1, ипдиотесд, 


Ох-теричная система счисления 


Можно ли вставить строку прямо в запрос и не бояться уязвимостей? 


Ответ: да, если у вас М$ 50 


// Данные заэкранируем хексом - фича мсскл 
$$Ег1пд = '0х'. Б1п2Вех($уа\ие); 
$$491 = "ТМЗЕВТ ТМТО [табТе] ( [уагспаг_со1]) УАЕУЕЗ ($$Ег1п9)"; 


Ох-теричная система счисления 


Можно ли вставить строку прямо в запрос и не бояться уязвимостей? 


Ответ: да, если у вас М$ 50 


// Данные заэкранируем хексом - фича мсскл 
$$Ег1пд = '0х'. Б1п2Вех($уа\ие); 
$$491 = "ТМЗЕВТ ТМТО [табТе] ( [уагспаг_со1]) УАЕУЕЗ ($$Ег1п9)"; 


Ну и теперь сохраняем строковые данные в ВОВ поле М$ 501 


ТР (1$_$Ег1пд($уа\ие)) { 
гесигп пем Ехргез$зтоп ( 'СОМ\УЕКТ (УАВВТМАКУ (МАХ), '. 
('0х'. Б1п2Вех($уаше)) .')'); 
} 


Спасибо Андрею Рычкову за науку 


Ну его нафиг... 


Так ведь и выгореть можно. 


Вместо вывода 


Все преимущества ОКМ, мокрые от слёз разработчиков 


Код может быть абстрактным 

Нет нужды писать «сырые» запросы 
Порог вхождения гораздо ниже 

Код не слишком привязан к движку БД 


Теперь, если вам понадобится своя ОКМ — вы будете готовы 


Вместо вывода 


Все преимущества ОКМ, мокрые от слёз разработчиков 


Код может быть абстрактным 

Нет нужды писать «сырые» запросы 
Порог вхождения гораздо ниже 

Код не слишком привязан к движку БД 


Теперь, если вам понадобится своя ОКМ — вы будете готовы, наверное. 
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